#pragma once
//文件版本
#include "../common/util.hpp"
#include "../common/log.hpp"
#include <iostream>
#include <string>
#include <vector>
#include <unordered_map>
#include <fstream>
#include <cstdlib>
#include <cassert>
#include <algorithm>

// 根据题目list文件，加载所有的题目信息到内存中
// model: 主要用来和数据进行交互，对外提供访问数据的接口
namespace ns_model
{
    using namespace std;
    using namespace ns_log;
    using namespace ns_util;
    struct Question
    {
        std::string number; // 题目编号，唯一
        std::string title;  // 题目的标题
        std::string star;   // 难度: 简单 中等 困难
        int cpu_limit;      // 题目的时间要求(S)
        int mem_limit;      // 题目的空间要去(KB)
        std::string desc;   // 题目的描述
        std::string header; // 题目预设给用户在线编辑器的代码
        std::string tail;   // 题目的测试用例，需要和header拼接，形成完整代码
    };
    const std::string questins_list = "./questions/questions_list";
    const std::string questins_path = "./questions/";
    class Model
    {
    private:
        // 题号 : 题目细节
        unordered_map<string, Question> questions;

    public:
        Model()
        {
            assert(LoadQuestionList(questins_list));
        }
        bool LoadQuestionList(const string &question_list)
        {
            // 加载配置文件: questions/questions.list + 题目编号文件
            ifstream in(question_list);
            if (!in.is_open())
            {
                LOG(FATAL) << " 加载题库失败,请检查是否存在题库文件" << "\n";
                return false;
            }
            string line;
            while (getline(in, line))
            {
                vector<string> tokens;
                stringutil::splitstring(line, &tokens, " ");
                // 1 判断回文数 简单 1 30000
                if (tokens.size() != 5)
                {
                    LOG(WARNING) << "加载部分题目失败, 请检查文件格式" << "\n";
                    continue;
                }
                Question q;
                q.number = tokens[0];
                q.title = tokens[1];
                q.star = tokens[2];
                q.cpu_limit = atoi(tokens[3].c_str());
                q.mem_limit = atoi(tokens[4].c_str());
                string path = questins_path;
                path += q.number;
                path += "/";
                fileutil::ReadFile(path + "desc.txt", &(q.desc), true);
                fileutil::ReadFile(path + "header.cpp", &(q.header), true);
                fileutil::ReadFile(path + "tail.cpp", &(q.tail), true);
                questions.insert({q.number, q});
            }
            LOG(INFO) << "加载题库...成功!" << "\n";
            in.close();
            return true;
        }
        //拿到所有题目
        bool GetAllQuestions(vector<Question> *out)
        {
            if (questions.size() == 0)
            {
                LOG(ERROR) << "用户获取题库失败" << "\n";
                return false;
            }
            for (const auto &q : questions)
            {
                out->push_back(q.second); // first: key, second: value
            }
             
            return true;
        }
        //拿到一道题的题目细节
        bool GetOneQuestion(const std::string &number, Question *q)
        {
            const auto &iter = questions.find(number);
            if (iter == questions.end())
            {
                LOG(ERROR) << "用户获取题目失败, 题目编号: " << number << "\n";
                return false;
            }
            (*q) = iter->second;
            return true;
        }
        ~Model()
        {
        }
    };
}

// #pragma once

// #include<iostream>
// #include<string>
// #include<vector>
// #include<unordered_map>
// #include<assert.h>

// namespace ns_model
// {
//     using namespace std;

//     struct question
//     {
//         std::string number; //题目编号，唯一
//         std::string title; //题目的标题
//         std::string star; //难度: 简单 中等 困难
//         int cpu_limit; //题目的时间要求(S)
//         int mem_limit; //题目的空间要去(KB)
//         std::string desc; //题目的描述
//         std::string header; //题目预设给用户在线编辑器的代码
//         std::string tail; //题目的测试用例，需要和header拼接，形成完整代码
//     };

//     class model
//     {
//         private:
//         //题目列表
//         unordered_map<std::string,question> _questions;

//         public:
//         model()
//         {
//             assert(LoadQuertionList());
//         }
//         //加载题目列表
//         bool LoadQuertionList()
//         {
//             //todo
//             return true;
//         }
//         //获取题目列表
//         bool GetQuestionList(vector<question> *out)
//         {
//             if(_questions.size() == 0)
//             {
//                 return false;
//             }
//             for(const auto &q:_questions)
//             {
//                 (*out).push_back(q.second);
//             }
//             return true;
//         }
//         //获取单个题目信息
//         bool GetOneQuestion(const std::string &number,question *out)
//         {
//             auto iter  =_questions.find(number);
//             if(iter == _questions.end())
//             {
//                 return false;
//             }
//             (*out) = iter->second;
//             return true;
//         }
//         ~model()
//         {}
//     };
// }
